import random
import re
from datetime import datetime

from flask import abort, jsonify
from flask import current_app
from flask import json
from flask import make_response
from flask import session

from info import constants, db
from info import redis_store
from info.models import User
from info.utils.response_code import RET
from sms import CCP
from . import passport_blu
from flask import request
from info.utils.captcha.captcha import  captcha

@passport_blu.route("/logout")
def logout():
    #退出登录
    #pop是tqbwsession中的数据(dict)
    #pop会有一个返回值没有就是None
    session.pop("user_id",None)
    session.pop("mobile", None)
    session.pop("nick_name", None)
    session.pop("is_admin",None)
    return jsonify(errno = RET.OK,errmsg = "退出成功")

@passport_blu.route("/login",methods = ["POST"])
def login():
    """
    1.获取参数
    2.校验参数
    3.校验密码是否正确
    4.保存用户的登录状态
    5.响应
    :return:
    """
    #1.获取参数
    params_dict = request.json
    mobile = params_dict.get("mobile")
    passport = params_dict.get("passport")
    #2.校验参数
    if not all([mobile,passport]):
        return jsonify(errno = RET.PARAMERR,errmsg = "参数错误")
    #校验手机号是否正确
    if not re.match("1[35678]\\d{9}",mobile):
        return jsonify(errno = RET.PARAMERR,errmsg = "手机号格式不正确")
    #3.校验密码更不正确
    #先查询出当前是否有指定手机号的用户
    try:
        user = User.query.filter(User.mobile == mobile).first()
    except Exception as e :
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg = "数据查询错误")
    #判断用户是否存在
    if not user:
        return jsonify(errno=RET.NODATA,errmsg = "用户不存在")
    #校验登录的密码和当前用户的密码是否一致
    if not user.check_password(passport):
        return jsonify(errno = RET.PWDERR,errmsg = "用户名或者密码错误")
    #4.保存用户的登录状态
    session["user_id"] = user.id
    session["mobile"] = user.mobile
    session["nick_name"]  = user.nick_name
    #调协当前用户最后一次登录的时间
    user.last_login = datetime.now()
    #如果在视图函数中，对模型身上的属性有修改，那么fmsvcommit到数据库保存
    #但是其实可以不用自己去写db.session.commit(),前提是对SQLAlchemy有过相关配置
    # try:
    #     db.session.commit()
    # except Exception as e :
    #     db.session.rollback()
    #     current_app.logger.error(e)
    #5.响应
    return jsonify(errno = RET.OK,errmsg = "登录成功")



@passport_blu.route('/register',methods = ["POST"])
def register():
    """
    注册的逻辑
    1.获取参数
    2.校验参数
    3.取到服务器保存的真实的短信验证码内容
    4.校验用户输入的短信验证码内容和真实验证码内容是否一致
    5.如果一致，初始化User模型，并且属性
    6.将user模型添加数据库
    7.返回响应
    :return:
    """
    #1.获取参数
    param_dict = request.json
    mobile = param_dict.get("mobile")
    smscode = param_dict.get("smscode")
    password = param_dict.get("password")
    #2.校验参数
    if not all([mobile,smscode,password]):
        return jsonify(errno=RET.PARAMERR,errmsg = "参数")
    #校验手机号是否正确
    if not re.match("1[35678]\\d{9}",mobile):
        return jsonify(errno = RET.PARAMERR,errmsg = "手机号格式不正确")
    #3.取到服务器保存的真实的短信验证码内容
    try:
        real_sms_code = redis_store.get("SMS_"+mobile)
    except Exception as e :
        current_app.logger.error(e)
        return jsonify(errno = RET.DBERR,errmsg = "数据查询失败")
    if not real_sms_code:
        return jsonify(errno = RET.DATAERR,errmsg = "验证码已过期")
    #4.校验用户输入的短信验证码内容和真实验证码内容是否一致
    if real_sms_code != smscode:
        return jsonify(errno = RET.DATAERR,errmsg = "验证码输输入错误")
    #5.如果一致，初始化User模型，并且赋值属性
    user = User()
    user.mobile = mobile
    #暂时没有昵称，使用手机号代替
    user.nick_name = mobile
    #记录用户最后一次登录时间
    user.last_login = datetime.now()
    #对密码做处理
    #需求：在设置password的时候，去对password进行加密,并且将加密结果给user.password_hash赋值
    user.password = password



    #6.添加到数据库
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as e :
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(erron=RET.DBERR,errmsg = "数据保存失败")
    #信session中保存数据表示当前已经登录
    session["user_id"] = user.id
    session["mobile"] = user.mobile
    session["nick_name"]  = user.nick_name
    #7.返回响应
    return jsonify(errno = RET.OK,errmsg = "注册成功")

@passport_blu.route('/sms_code',methods = ["POST"])
def send_sms_code():
    """
    1.获取参数：手机号，图片验证码内容，图片验证码的编号（随机值）
    2.校验参数（参数是否符合规则，判断是否有值）
    3.先从redis中取出真实的验证码内容
    4.与用户的验证码内容进行对比，如果对比不一致，那么返回验证码输入错误
    5.如果一致，生成验证码的内容（随机数据）
    6.发送短信验主证码
    7.告知发送结果
    :return:
    """

    '{"mobiel":"18111111","image_code":"aaaa","image_code_id":"f16s5d1fw1fw16f561sa"}'
    #1.获取参数：手机号，图片验证码内容，图片验证码的编号（随机值）
    # params_dict = json.loads(request.data)
    params_dict = request.json

    mobile  = params_dict.get("mobile")
    image_code = params_dict.get("image_code")
    image_code_id = params_dict.get("image_code_id")

    # 2.校验参数（参数是否符合规则，判断是否有值
    #判断参数是否有值
    if not all([mobile,image_code,image_code_id]):
        # return jsonify(errno = "4100",errmsg = "参数有误")
        return jsonify(errno=RET.PARAMERR, errmsg="参数有误")
    #校验手机号是否正确
    if not re.match("1[35678]\\d{9}",mobile):
        return jsonify(errno = RET.PARAMERR,errmsg = "手机号格式不正确")
    #3.先从redis中取出真实的验证码内容
    try:
        real_image_code = redis_store.get("ImageCodeId_"+image_code_id)
    except Exception as e:
        current_app.looger.error(e)
        return jsonify(errno = RET.DBERR,errmsg = '数据查询失败')
    if not real_image_code:
        return jsonify(errno = RET.NODATA,errmsg = "图片验证码已过期")
    #4.与用户的验证码内容进行对比，如果对比不一致，那么返回验证码输入错误
    if real_image_code.upper() != image_code.upper():
        return jsonify(errno = RET.DATAERR,errmsg = "验证码输入错误")
    #5.如果一致，生成短信验证码的内容（随机数据）
    #随机数字，保证数字长度为6位，不够在前面补上0
    sms_code_str = "%06d"%random.randint(0,999999)
    current_app.logger.debug("短信验证码内容是:%s"%sms_code_str)
    #6.发送短信验证码
    result = CCP().send_template_sms(mobile,[sms_code_str,constants.SMS_CODE_REDIS_EXPIRES/5],"1")
    print(result)

    if result != 0:
        #代表发送不成功
        return jsonify(errno=RET.THIRDERR,errmsg = "发送短信失败")
    #保存验证码内容到redis
    try:
        redis_store.set("SMS_"+mobile,sms_code_str,constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno = RET.DBERR,errmsg = "数据保存")

    #7.告知发送结果
    return jsonify(errno = RET.OK,errmsg = "发送成功")


@passport_blu.route('/image_code')
def get_image_code():
    """
    生成图片验证码并返回
    1.取到参数
    2.判断参数是否有值
    3.生成图片验证码
    4.保存图片验证码文字内容到redis
    5.返回验证码图片
    :return:
    """
    #1取到参数
    #args : 取到url中？后面的参数
    image_code_id = request.args.get("imageCodeId",None)
    #2判断参数是否有值
    if not image_code_id:
        return abort(403)
    #3生成图片验证码
    name,text,image = captcha.generate_captcha()
    #4保存图片验证码文字内容到redis
    try:
        redis_store.set("ImageCodeId_" + image_code_id,text,constants.IMAGE_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        abort(500)
    #5.返回验证码图片
    response = make_response(image)
    #设置数据的类型，以便浏览器更加智能识别其是什么类型
    response.headers["Content-Type"] = "image/jpg"
    return response





